Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat/generic-oidc-claims: add the ability to set email and id_from_idp from separate OIDC claims in the generic OIDC idp #1153

Open
wants to merge 7 commits into
base: master
Choose a base branch
from

Conversation

davidmonro
Copy link

Link to JIRA ticket if there is one:

New Features

Allow returning email and id_from_idp claims from the generic OIDC get_auth_info() code, as well as the user_id.

Breaking Changes

Bug Fixes

Improvements

Dependency updates

Deployment changes

Notes:

This is the first time I've attempted to contribute to gen3, so I may well be doing it wrong!
I have tested this locally using the dex idp https://github.com/dexidp/dex as the OIDC server

fence/resources/openid/idp_oauth2.py Outdated Show resolved Hide resolved
self.logger.exception(
f"Can't get {field} from claims: {claims}"
)
return {"error": f"Can't get {field} from claims"}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We shouldn't return an error if the email field is missing. The claims come from id_token. Throwing an error because we get an id_token would effectively require the email claim in the id_token. This would take fence out of spec with OIDC standard. See here https://openid.net/specs/openid-connect-core-1_0.html#IDToken

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've reworked this, but the new version doesn't return an error if all the fields are missing, which probably isn't right either. I'm not sure what the right answer is - should I check if the requested fields are mandatory, or throw an error if the field requested for user_id_field doesn't exist but ignore the others?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should only throw an error if the fields required by the OIDC spec are missing. I believe this is already covered by Fence so there should be no new errors that are thrown unless there's specific configuration that requires email verification (if I understand this PR correctly).

return {"error": f"Can't get {field} from claims"}

# Field is email, but isn't verified and we aren't assuming all emails are verified
if field == "email" and not (claims.get("email_verified") or self.settings.get("assume_emails_verified")):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

test_idp_oauth2 need to be updated to test the branch conditions here to ensure that the error is returned as expected depending on the use of "assume_emails_verified"

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This turns out to be rather difficult, at least with the code as it is now, because get_auth_info() takes code as an argument that it wants to extract the claims from, and having done some experiments I'm not sure the mock idp can handle that?

I could refactor the claims parsing out to another function that would then be more easily testable if you'd like.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You should be able to mock the response of get_jwt_claims_identity, so its not reliant on code. Either way, this is a pretty critical part of our authentication service so tests that ensure that this change doesn't break the default case and works as expected in your case are valuable.

return {"error": f"Can't get {field} from claims"}

# Field is email, but isn't verified and we aren't assuming all emails are verified
if field == "email" and not (claims.get("email_verified") or self.settings.get("assume_emails_verified")):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

assumed_email_verified needs to be added to config-default.yaml under OPENID_CONNECT with documentation explaining its usage for the respective IDPs of interest.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Documented in 7951b84

fence/resources/openid/idp_oauth2.py Outdated Show resolved Hide resolved
k-burt-uch
k-burt-uch previously approved these changes Nov 4, 2024
email_field: '' # optional (default "email"); claims field to get the user email from
# If your IDP doesn't support the email_verified claim, set this to True to
# accept the email address claim anyway
assume_emails_verified: False
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

required_email_verified makes more sense here. It should default to false so we don't throw this new requirement unexpectedly on existing fence deployments and disrupt login.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I reused the existing setting from the cognito code. Happy to refactor this if you'd like.
Also, see #1153 (comment)

if claims.get(field):

# Field is email, but isn't verified and we aren't assuming all emails are verified
if field == "email" and not (claims.get("email_verified") or self.settings.get("assume_emails_verified")):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should, be default, assume that email verification isn't a requirement. There is not a requirement in the OAuth2 or OIDC spec that requires email verification so this will break standards in our authentication service.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The original code would always return an error in the case that the email_verified claim wasn't present, and didn't give you a way to disable that, so I haven't changed the default behaviour here.

Again, happy to tweak that if desired.

self.logger.exception(
f"Can't get {field} from claims: {claims}"
)
return {"error": f"Can't get {field} from claims"}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should only throw an error if the fields required by the OIDC spec are missing. I believe this is already covered by Fence so there should be no new errors that are thrown unless there's specific configuration that requires email verification (if I understand this PR correctly).

return {"error": f"Can't get {field} from claims"}

# Field is email, but isn't verified and we aren't assuming all emails are verified
if field == "email" and not (claims.get("email_verified") or self.settings.get("assume_emails_verified")):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You should be able to mock the response of get_jwt_claims_identity, so its not reliant on code. Either way, this is a pretty critical part of our authentication service so tests that ensure that this change doesn't break the default case and works as expected in your case are valuable.

@k-burt-uch k-burt-uch dismissed their stale review November 4, 2024 18:04

Mistaken approval

Copy link
Contributor

@k-burt-uch k-burt-uch left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Overall looking better. We need this to be disabled by default and some tests to verify this works as expected.

@michaelfitzo
Copy link

Hello! Before moving ahead with any final approvals, the engineering team feels that it would be extremely useful to us to hear a bit more background on the request. We have recently implemented a requirement that PRs must typically include a Community Feature Document, which gives us a bit more background on the request, your requirements, and use cases.

https://docs.google.com/document/d/1P2dfqnSH-e7OX1Hw62sDL8zcR7gZp4d152TlDBlomDc/edit?usp=sharing

The document looks long, but I would ask you to focus on the requirements and use cases sections along with whatever else you can. Apologies again for not sending sooner, but I think this will unblock any final reviews required. Let me know if you have any questions about this

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants